/**
 * Copyright (c) Microsoft Corporation.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

import * as mcp from '../sdk/exports';
import { defaultConfig } from '../browser/config';
import { BrowserServerBackend } from '../browser/browserServerBackend';
import { Tab } from '../browser/tab';
import { stripAnsiEscapes } from '../../util';
import { identityBrowserContextFactory } from '../browser/browserContextFactory';

import type * as playwright from '../../../index';
import type { Page } from '../../../../playwright-core/src/client/page';
import type { TestInfo } from '../../../test';

export type BrowserMCPRequest = {
  initialize?: { clientInfo: mcp.ClientInfo },
  listTools?: {},
  callTool?: { name: string, arguments: mcp.CallToolRequest['params']['arguments'] },
  close?: {},
};

export type BrowserMCPResponse = {
  initialize?: { pausedMessage: string },
  listTools?: mcp.Tool[],
  callTool?: mcp.CallToolResult,
  close?: {},
};

export function createCustomMessageHandler(testInfo: TestInfo, context: playwright.BrowserContext) {
  let backend: BrowserServerBackend | undefined;
  return async (data: BrowserMCPRequest): Promise<BrowserMCPResponse> => {
    if (data.initialize) {
      if (backend)
        throw new Error('MCP backend is already initialized');
      backend = new BrowserServerBackend({ ...defaultConfig, capabilities: ['testing'] }, identityBrowserContextFactory(context));
      await backend.initialize(data.initialize.clientInfo);
      const pausedMessage = await generatePausedMessage(testInfo, context);
      return { initialize: { pausedMessage } };
    }

    if (data.listTools) {
      if (!backend)
        throw new Error('MCP backend is not initialized');
      return { listTools: await backend.listTools() };
    }

    if (data.callTool) {
      if (!backend)
        throw new Error('MCP backend is not initialized');
      return { callTool: await backend.callTool(data.callTool.name, data.callTool.arguments) };
    }

    if (data.close) {
      backend?.serverClosed();
      backend = undefined;
      return { close: {} };
    }

    throw new Error('Unknown MCP request');
  };
}

async function generatePausedMessage(testInfo: TestInfo, context: playwright.BrowserContext) {
  const lines: string[] = [];

  if (testInfo.errors.length) {
    lines.push(`### Paused on error:`);
    for (const error of testInfo.errors)
      lines.push(stripAnsiEscapes(error.message || ''));
  } else {
    lines.push(`### Paused at end of test. ready for interaction`);
  }

  for (let i = 0; i < context.pages().length; i++) {
    const page = context.pages()[i];
    const stateSuffix = context.pages().length > 1 ? (i + 1) + ' of ' + (context.pages().length) : 'state';
    lines.push(
        '',
        `### Page ${stateSuffix}`,
        `- Page URL: ${page.url()}`,
        `- Page Title: ${await page.title()}`.trim()
    );
    // Only print console errors when pausing on error, not when everything works as expected.
    let console = testInfo.errors.length ? await Tab.collectConsoleMessages(page) : [];
    console = console.filter(msg => msg.type === 'error');
    if (console.length) {
      lines.push('- Console Messages:');
      for (const message of console)
        lines.push(`  - ${message.toString()}`);
    }
    lines.push(
        `- Page Snapshot:`,
        '```yaml',
        (await (page as Page)._snapshotForAI()).full,
        '```',
    );
  }

  lines.push('');
  if (testInfo.errors.length)
    lines.push(`### Task`, `Try recovering from the error prior to continuing`);

  return lines.join('\n');
}
